home *** CD-ROM | disk | FTP | other *** search
/ Chip 2007 January, February, March & April / Chip-Cover-CD-2007-02.iso / Pakiet internetowy / Przegladarki internetowe / Mozilla Seamonkey 1.0.5 pl / seamonkey-1.0.5.pl-PL.win32.installer.exe / CHATZILLA.XPI / bin / chrome / chatzilla.jar / content / chatzilla / config.js < prev    next >
Encoding:
Text File  |  2005-04-27  |  55.9 KB  |  1,738 lines

  1. /* ***** BEGIN LICENSE BLOCK *****
  2.  * Version: MPL 1.1/GPL 2.0/LGPL 2.1
  3.  *
  4.  * The contents of this file are subject to the Mozilla Public License Version
  5.  * 1.1 (the "License"); you may not use this file except in compliance with
  6.  * the License. You may obtain a copy of the License at
  7.  * http://www.mozilla.org/MPL/
  8.  *
  9.  * Software distributed under the License is distributed on an "AS IS" basis,
  10.  * WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License
  11.  * for the specific language governing rights and limitations under the
  12.  * License.
  13.  *
  14.  * The Original Code is ChatZilla.
  15.  *
  16.  * The Initial Developer of the Original Code is James Ross.
  17.  * Portions created by the Initial Developer are Copyright (C) 2004
  18.  * the Initial Developer. All Rights Reserved.
  19.  *
  20.  * Contributor(s):
  21.  *   James Ross <silver@warwickcompsoc.co.uk>
  22.  *
  23.  * Alternatively, the contents of this file may be used under the terms of
  24.  * either the GNU General Public License Version 2 or later (the "GPL"), or
  25.  * the GNU Lesser General Public License Version 2.1 or later (the "LGPL"),
  26.  * in which case the provisions of the GPL or the LGPL are applicable instead
  27.  * of those above. If you wish to allow use of your version of this file only
  28.  * under the terms of either the GPL or the LGPL, and not to allow others to
  29.  * use your version of this file under the terms of the MPL, indicate your
  30.  * decision by deleting the provisions above and replace them with the notice
  31.  * and other provisions required by the GPL or the LGPL. If you do not delete
  32.  * the provisions above, a recipient may use your version of this file under
  33.  * the terms of any one of the MPL, the GPL or the LGPL.
  34.  *
  35.  * ***** END LICENSE BLOCK ***** */
  36.  
  37. const MEDIATOR_CONTRACTID   = "@mozilla.org/appshell/window-mediator;1";
  38. const FILEPICKER_CONTRACTID = "@mozilla.org/filepicker;1";
  39.  
  40. const nsIWindowMediator     = Components.interfaces.nsIWindowMediator;
  41. const nsIFilePicker         = Components.interfaces.nsIFilePicker;
  42.  
  43. const CONFIG_WINDOWTYPE     = "irc:chatzilla:config";
  44.  
  45. /* Now we create and set up some required items from other Chatzilla JS files 
  46.  * that we really have no reason to load, but the ones we do need won't work 
  47.  * without these...
  48.  */
  49. var ASSERT = function(cond, msg) { if (!cond) { alert(msg); } return cond; }
  50. var client;
  51.  
  52. function CIRCNetwork() {}
  53. function CIRCServer() {}
  54. function CIRCChannel() {}
  55. function CIRCChanUser() {}
  56.  
  57. function getObjectDetails(obj)
  58. {
  59.     var rv = new Object();
  60.     rv.sourceObject = obj;
  61.     rv.TYPE = obj.TYPE;
  62.     rv.parent = ("parent" in obj) ? obj.parent : null;
  63.     rv.user = null;
  64.     rv.channel = null;
  65.     rv.server = null;
  66.     rv.network = null;
  67.     
  68.     switch (obj.TYPE)
  69.     {
  70.         case "PrefNetwork":
  71.             rv.network = obj;
  72.             if ("primServ" in rv.network)
  73.                 rv.server = rv.network.primServ;
  74.             else
  75.                 rv.server = null;
  76.             break;
  77.             
  78.         case "PrefChannel":
  79.             rv.channel = obj;
  80.             rv.server = rv.channel.parent;
  81.             rv.network = rv.server.parent;
  82.             break;
  83.             
  84.         case "PrefUser":
  85.             rv.user = obj;
  86.             rv.server = rv.user.parent;
  87.             rv.network = rv.server.parent;
  88.             break;
  89.     }
  90.     
  91.     return rv;
  92. }
  93.  
  94. /* Global object for the prefs. The 'root' of all the objects to do with the
  95.  * prefs.
  96.  */
  97. function PrefGlobal()
  98. {
  99.     this.networks = new Object();
  100.     this.commandManager = new Object();
  101.     this.commandManager.defineCommand = function() {};
  102.     this.entities = new Object();
  103. }
  104. PrefGlobal.prototype.TYPE = "PrefGlobal";
  105.  
  106. /* Represents a single network in the hierarchy.
  107.  * 
  108.  * |force| - If true, sets a pref on this object. This makes sure the object
  109.  *           is "known" next time we load up (since we look for any prefs).
  110.  * 
  111.  * |show|  - If true, the object still exists even if the magic pref is not set.
  112.  *           Thus, allows an object to exist without any prefs set.
  113.  */
  114. function PrefNetwork(parent, name, force, show)
  115. {
  116.     if (name in parent.networks)
  117.         return parent.networks[name];
  118.     
  119.     this.parent = parent;
  120.     this.unicodeName = name;
  121.     this.viewName = name;
  122.     this.canonicalName = name;
  123.     this.encodedName = name;
  124.     this.prettyName = getMsg(MSG_PREFS_FMT_DISPLAY_NETWORK, this.unicodeName);
  125.     this.servers = new Object();
  126.     this.primServ = new PrefServer(this, "dummy server");
  127.     this.channels = this.primServ.channels;
  128.     this.users = this.primServ.users;
  129.     this.prefManager = getNetworkPrefManager(this);
  130.     this.prefs = this.prefManager.prefs;
  131.     this.prefManager.onPrefChanged = function(){};
  132.     
  133.     if (force)
  134.         this.prefs["hasPrefs"] = true;
  135.     
  136.     if (this.prefs["hasPrefs"] || show)
  137.         this.parent.networks[this.canonicalName] = this;
  138.     
  139.     return this;
  140. };
  141. PrefNetwork.prototype.TYPE = "PrefNetwork";
  142.  
  143. /* Cleans up the mess. */
  144. PrefNetwork.prototype.clear =
  145. function pnet_clear()
  146. {
  147.     this.prefs["hasPrefs"] = false;
  148.     delete this.parent.networks[this.canonicalName];
  149. }
  150.  
  151. /* A middle-management object.
  152.  * 
  153.  * Exists only to satisfy the IRC library pref functions that expect this
  154.  * particular hierarchy.
  155.  */
  156. function PrefServer(parent, name)
  157. {
  158.     this.parent = parent;
  159.     this.unicodeName = name;
  160.     this.viewName = name;
  161.     this.canonicalName = name;
  162.     this.encodedName = name;
  163.     this.prettyName = this.unicodeName; // Not used, thus not localised.
  164.     this.channels = new Object();
  165.     this.users = new Object();
  166.     this.parent.servers[name] = this;
  167.     return this;
  168. };
  169. PrefServer.prototype.TYPE = "PrefServer";
  170.  
  171. /* Represents a single channel in the hierarchy.
  172.  * 
  173.  * |force| and |show| the same as PrefNetwork.
  174.  */
  175. function PrefChannel(parent, name, force, show)
  176. {
  177.     if (name in parent.channels)
  178.         return parent.channels[name];
  179.     
  180.     this.parent = parent;
  181.     this.unicodeName = name;
  182.     this.viewName = name;
  183.     this.canonicalName = name;
  184.     this.encodedName = name;
  185.     this.prettyName = getMsg(MSG_PREFS_FMT_DISPLAY_CHANNEL, 
  186.                              [this.parent.parent.unicodeName, this.unicodeName]);
  187.     this.prefManager = getChannelPrefManager(this);
  188.     this.prefs = this.prefManager.prefs;
  189.     this.prefManager.onPrefChanged = function(){};
  190.     
  191.     if (force)
  192.         this.prefs["hasPrefs"] = true;
  193.     
  194.     if (this.prefs["hasPrefs"] || show)
  195.         this.parent.channels[this.canonicalName] = this;
  196.     
  197.     return this;
  198. };
  199. PrefChannel.prototype.TYPE = "PrefChannel";
  200.  
  201. /* Cleans up the mess. */
  202. PrefChannel.prototype.clear =
  203. function pchan_clear()
  204. {
  205.     this.prefs["hasPrefs"] = false;
  206.     delete this.parent.channels[this.canonicalName];
  207. }
  208.  
  209. /* Represents a single user in the hierarchy.
  210.  * 
  211.  * |force| and |show| the same as PrefNetwork.
  212.  */
  213. function PrefUser(parent, name, force, show)
  214. {
  215.     if (name in parent.users)
  216.         return parent.users[name];
  217.     
  218.     this.parent = parent;
  219.     this.unicodeName = name;
  220.     this.viewName = name;
  221.     this.canonicalName = name;
  222.     this.encodedName = name;
  223.     this.prettyName = getMsg(MSG_PREFS_FMT_DISPLAY_USER, 
  224.                              [this.parent.parent.unicodeName, this.unicodeName]);
  225.     this.prefManager = getUserPrefManager(this);
  226.     this.prefs = this.prefManager.prefs;
  227.     this.prefManager.onPrefChanged = function(){};
  228.     
  229.     if (force)
  230.         this.prefs["hasPrefs"] = true;
  231.     
  232.     if (this.prefs["hasPrefs"] || show)
  233.         this.parent.users[this.canonicalName] = this;
  234.     
  235.     return this;
  236. };
  237. PrefUser.prototype.TYPE = "PrefUser";
  238.  
  239. /* Cleans up the mess. */
  240. PrefUser.prototype.clear =
  241. function puser_clear()
  242. {
  243.     this.prefs["hasPrefs"] = false;
  244.     delete this.parent.users[this.canonicalName];
  245. }
  246.  
  247. // Stores a list of |PrefObject|s.
  248. function PrefObjectList()
  249. {
  250.     this.objects = new Array();
  251.     
  252.     return this;
  253. }
  254.  
  255. // Add an object, and init it's private data.
  256. PrefObjectList.prototype.addObject =
  257. function polist_addObject(pObject)
  258. {
  259.     this.objects.push(pObject);
  260.     return pObject.privateData = new ObjectPrivateData(pObject, this.objects.length - 1);
  261. }
  262.  
  263. /* Removes an object, without changing the index. */
  264. PrefObjectList.prototype.deleteObject =
  265. function polist_addObject(index)
  266. {
  267.     this.objects[index].privateData.clear();
  268.     this.objects[index].clear();
  269.     this.objects[index] = { privateData: null };
  270. }
  271.  
  272. // Get a specific object.
  273. PrefObjectList.prototype.getObject =
  274. function polist_getObject(index)
  275. {
  276.     return this.objects[index].privateData;
  277. }
  278.  
  279. // Gets the private data for an object.
  280. PrefObjectList.getPrivateData =
  281. function polist_getPrivateData(object)
  282. {
  283.     return object.privateData;
  284. }
  285.  
  286. // Stores the pref object's private data.
  287. function ObjectPrivateData(parent, index)
  288. {
  289.     this.parent = parent;  // Real pref object.
  290.     this.prefs = new Object();
  291.     this.groups = new Object();
  292.     
  293.     this.arrayIndex = index;
  294.     this.deckIndex = -1;
  295.     this.dataLoaded = false;
  296.     
  297.     var treeObj = document.getElementById("pref-tree-object");
  298.     this.tree = document.getElementById("pref-tree");
  299.     this.treeContainer = document.createElement("treeitem");
  300.     this.treeNode = document.createElement("treerow");
  301.     this.treeCell = document.createElement("treecell");
  302.     
  303.     this.treeContainer.setAttribute("prefobjectindex", this.arrayIndex);
  304.     this.treeCell.setAttribute("label", this.parent.unicodeName);
  305.     
  306.     switch (this.parent.TYPE)
  307.     {
  308.         case "PrefChannel":
  309.         case "PrefUser":
  310.             var p = this.parent.parent.parent;  // Network object.
  311.             var pData = PrefObjectList.getPrivateData(p);
  312.             
  313.             if (!("treeChildren" in pData) || !pData.treeChildren)
  314.             {
  315.                 pData.treeChildren = document.createElement("treechildren");
  316.                 pData.treeContainer.appendChild(pData.treeChildren);
  317.                 treeObj.view.toggleOpenState(treeObj.view.rowCount - 1);
  318.             }
  319.             pData.treeContainer.setAttribute("container", "true");
  320.             pData.treeChildren.appendChild(this.treeContainer);
  321.             break;
  322.             
  323.         default:
  324.             this.tree.appendChild(this.treeContainer);
  325.             break;
  326.     }
  327.     
  328.     this.treeContainer.appendChild(this.treeNode);
  329.     this.treeNode.appendChild(this.treeCell);
  330.     
  331.     return this;
  332. }
  333.  
  334. // Creates all the XUL elements needed to show this pref object.
  335. ObjectPrivateData.prototype.loadXUL =
  336. function opdata_loadXUL(tabOrder)
  337. {
  338.     var t = this;
  339.     
  340.     /* Function that sorts the preferences by their label, else they look 
  341.      * fairly random in order.
  342.      * 
  343.      * Sort keys: not grouped, sub-group name, boolean, pref label.
  344.      */
  345.     function sortByLabel(a, b) {
  346.         if (t.prefs[a].subGroup || t.prefs[b].subGroup)
  347.         {
  348.             // Non-grouped go first.
  349.             if (!t.prefs[a].subGroup)
  350.                 return -1;
  351.             if (!t.prefs[b].subGroup)
  352.                 return 1;
  353.             
  354.             // Sub-group names.
  355.             if (t.prefs[a].subGroup < t.prefs[b].subGroup)
  356.                 return -1;
  357.             if (t.prefs[a].subGroup > t.prefs[b].subGroup)
  358.                 return 1;
  359.         }
  360.         
  361.         // Booleans go first.
  362.         if ((t.prefs[a].type == "boolean") && (t.prefs[b].type != "boolean"))
  363.             return -1;
  364.         if ((t.prefs[a].type != "boolean") && (t.prefs[b].type == "boolean"))
  365.             return 1;
  366.         
  367.         // ...then label.
  368.         if (t.prefs[a].label < t.prefs[b].label)
  369.             return -1;
  370.         if (t.prefs[a].label > t.prefs[b].label)
  371.             return 1;
  372.         return 0;
  373.     };
  374.     
  375.     if (this.deckIndex >= 0)
  376.         return;
  377.     
  378.     this.deck = document.getElementById("pref-object-deck");
  379.     this.tabbox = document.createElement("tabbox");
  380.     this.tabs = document.createElement("tabs");
  381.     this.tabPanels = document.createElement("tabpanels");
  382.     
  383.     this.tabbox.setAttribute("flex", 1);
  384.     this.tabPanels.setAttribute("flex", 1);
  385.     
  386.     this.tabbox.appendChild(this.tabs);
  387.     this.tabbox.appendChild(this.tabPanels);
  388.     this.deck.appendChild(this.tabbox);
  389.     
  390.     this.deckIndex = this.deck.childNodes.length - 1;
  391.     
  392.     this.loadData();
  393.     
  394.     var prefList = keys(this.prefs);
  395.     prefList.sort(sortByLabel);
  396.     
  397.     for (var i = 0; i < tabOrder.length; i++)
  398.     {
  399.         var pto = tabOrder[i];
  400.         var needTab = pto.fixed;
  401.         if (!needTab)
  402.         {
  403.             // Not a "always visible" tab, check we need it.
  404.             for (var j = 0; j < prefList.length; j++)
  405.             {
  406.                 if (this.prefs[prefList[j]].mainGroup == pto.name)
  407.                 {
  408.                     needTab = true;
  409.                     break;
  410.                 }
  411.             }
  412.         }
  413.         if (needTab)
  414.             this.addGroup(pto.name);
  415.     }
  416.     
  417.     for (i = 0; i < prefList.length; i++)
  418.         this.prefs[prefList[i]].loadXUL();
  419.     
  420.     if (this.tabs.childNodes.length > 0)
  421.         this.tabbox.selectedIndex = 0;
  422. }
  423.  
  424. // Loads all the prefs.
  425. ObjectPrivateData.prototype.loadData =
  426. function opdata_loadData()
  427. {
  428.     if (this.dataLoaded)
  429.         return;
  430.     
  431.     this.dataLoaded = true;
  432.     
  433.     // Now get the list of pref names, and add them...
  434.     var prefList = this.parent.prefManager.prefNames;
  435.     
  436.     for (i in prefList)
  437.         this.addPref(prefList[i]);
  438. }
  439.  
  440. // Clears up all the XUL objects and data.
  441. ObjectPrivateData.prototype.clear =
  442. function opdata_clear()
  443. {
  444.     //dd("Removing prefs for " + this.parent.displayName + " {");
  445.     if (!this.dataLoaded)
  446.         this.loadData();
  447.     for (var i in this.prefs)
  448.         this.prefs[i].clear();
  449.     //dd("}");
  450.     
  451.     if (this.deckIndex >= 0)
  452.     {
  453.         this.deck.removeChild(this.tabbox);
  454.         this.treeContainer.removeAttribute("container");
  455.         this.treeContainer.parentNode.removeChild(this.treeContainer);
  456.     }
  457. }
  458.  
  459. // Resets all the prefs to their original values.
  460. ObjectPrivateData.prototype.reset =
  461. function opdata_reset()
  462. {
  463.     for (var i in this.prefs)
  464.         if (this.prefs[i].type != "hidden")
  465.             this.prefs[i].reset();
  466. }
  467.  
  468. // Adds a pref to the internal data structures.
  469. ObjectPrivateData.prototype.addPref =
  470. function opdata_addPref(name)
  471. {
  472.     return this.prefs[name] = new PrefData(this, name);
  473. }
  474.  
  475. // Adds a group to a pref object's data.
  476. ObjectPrivateData.prototype.addGroup =
  477. function opdata_addPref(name)
  478. {
  479.     // Special group for prefs we don't want shown (nothing sinister here).
  480.     if (name == "hidden")
  481.         return null;
  482.     
  483.     if (!(name in this.groups))
  484.         this.groups[name] = new PrefMainGroup(this, name);
  485.     
  486.     return this.groups[name];
  487. }
  488.  
  489. // Represents a single pref on a single object within the pref window.
  490. function PrefData(parent, name)
  491. {
  492.     // We want to keep all this "worked out" info, so make a hash of all 
  493.     // the prefs on the pwData [Pref Window Data] property of the object.
  494.     
  495.     // First, lets find out what kind of pref we've got:
  496.     this.parent = parent;  // Private data for pref object.
  497.     this.name = name;
  498.     this.manager = this.parent.parent.prefManager;   // PrefManager.
  499.     this.record = this.manager.prefRecords[name];    // PrefRecord.
  500.     this.def = this.record.defaultValue;             // Default value.
  501.     this.type = typeof this.def;                     // Pref type.
  502.     this.val = this.manager.prefs[name];             // Current value.
  503.     this.startVal = this.val;                        // Start value.
  504.     this.label = this.record.label;                  // Display name.
  505.     this.help = this.record.help;                    // Help text.
  506.     this.group = this.record.group;                  // Group identifier.
  507.     this.labelFor = "none";                          // Auto-grouped label.
  508.     
  509.     // Handle defered prefs (call defer function, and use resulting 
  510.     // value/type instead).
  511.     if (this.type == "function")
  512.         this.def = this.def(this.name);
  513.     this.type = typeof this.def;
  514.     
  515.     // And those arrays... this just makes our life easier later by having 
  516.     // a particular name for array prefs.
  517.     if (this.def instanceof Array)
  518.         this.type = "array";
  519.     
  520.     if (this.group == "hidden")
  521.         this.type = "hidden";
  522.     
  523.     // Convert "a.b" into sub-properties...
  524.     var m = this.group.match(/^([^.]*)(\.(.*))?$/)
  525.     ASSERT(m, "Failed group match!");
  526.     this.mainGroup = m[1];
  527.     this.subGroup = m[3];
  528.     
  529.     return this;
  530. }
  531.  
  532. /* Creates all the XUL elements to display this one pref. */
  533. PrefData.prototype.loadXUL =
  534. function pdata_loadXUL()
  535. {
  536.     if (this.type == "hidden")
  537.         return;
  538.     
  539.     // Create the base box for the pref.
  540.     this.box = document.createElement("box");
  541.     this.box.orient = "horizontal";
  542.     this.box.setAttribute("align", "center");
  543.     
  544.     switch (this.type)
  545.     {
  546.         case "string":
  547.             label = document.createElement("label");
  548.             label.setAttribute("value", this.label);
  549.             label.width = 100;
  550.             label.flex = 1;
  551.             this.box.appendChild(label);
  552.             
  553.             this.edit = document.createElement("textbox");
  554.             // We choose the size based on the length of the default.
  555.             if (this.def.length < 8)
  556.                 this.edit.setAttribute("size", "10");
  557.             else if (this.def.length < 20)
  558.                 this.edit.setAttribute("size", "25");
  559.             else
  560.                 this.edit.flex = 1;
  561.             
  562.             var editCont = document.createElement("hbox");
  563.             editCont.flex = 1000;
  564.             editCont.appendChild(this.edit);
  565.             this.box.appendChild(editCont);
  566.             
  567.             // But if it's a file/URL...
  568.             if (this.def.match(/^([a-z]+:\/|[a-z]:\\)/i))
  569.             {
  570.                 // ...we make it as big as possible.
  571.                 this.edit.removeAttribute("size");
  572.                 this.edit.flex = 1;
  573.                 
  574.                 if (!this.name.match(/path$/i) && 
  575.                     (this.def.match(/^(file|chrome):\//i) ||
  576.                      this.name.match(/filename$/i)))
  577.                 {
  578.                     // So long as the pref name doesn't end in "path", and 
  579.                     // it's chrome:, file: or a local file, we add the button.
  580.                     var ext = "";
  581.                     var m = this.def.match(/\.([a-z0-9]+)$/);
  582.                     if (m)
  583.                         ext = "*." + m[1];
  584.                     
  585.                     // We're cheating again here, if it ends "filename" it's 
  586.                     // a local file path.
  587.                     var type = (this.name.match(/filename$/i) ? "file" : "fileurl");
  588.                     appendButton(this.box, "onPrefBrowse", 
  589.                                  { label: MSG_PREFS_BROWSE, spec: ext, 
  590.                                    kind: type });
  591.                 }
  592.             }
  593.             break;
  594.             
  595.         case "number":
  596.             label = document.createElement("label");
  597.             label.setAttribute("value", this.label);
  598.             label.width = 100;
  599.             label.flex = 1;
  600.             this.box.appendChild(label);
  601.             
  602.             this.edit = document.createElement("textbox");
  603.             this.edit.setAttribute("size", "5");
  604.             
  605.             editCont = document.createElement("hbox");
  606.             editCont.flex = 1000;
  607.             editCont.appendChild(this.edit);
  608.             this.box.appendChild(editCont);
  609.             break;
  610.             
  611.         case "boolean":
  612.             this.edit = document.createElement("checkbox");
  613.             this.edit.setAttribute("label", this.label);
  614.             this.box.appendChild(this.edit);
  615.             break;
  616.             
  617.         case "array":
  618.             this.box.removeAttribute("align");
  619.             
  620.             var oBox = document.createElement("box");
  621.             oBox.orient = "vertical";
  622.             oBox.flex = 1;
  623.             this.box.appendChild(oBox);
  624.             
  625.             if (this.help)
  626.             {
  627.                 label = document.createElement("label");
  628.                 label.appendChild(document.createTextNode(this.help));
  629.                 oBox.appendChild(label);
  630.             }
  631.             
  632.             this.edit = document.createElement("listbox");
  633.             this.edit.flex = 1;
  634.             this.edit.setAttribute("style", "height: 1em;");
  635.             this.edit.setAttribute("kind", "url");
  636.             if (this.def.length > 0 && this.def[0].match(/^file:\//))
  637.                 this.edit.setAttribute("kind", "fileurl");
  638.             this.edit.setAttribute("onselect", "gPrefWindow.onPrefListSelect(this);");
  639.             this.edit.setAttribute("ondblclick", "gPrefWindow.onPrefListEdit(this);");
  640.             oBox.appendChild(this.edit);
  641.             
  642.             var box = document.createElement("box");
  643.             box.orient = "vertical";
  644.             this.box.appendChild(box);
  645.             
  646.             // NOTE: This order is important - getRelatedItem needs to be 
  647.             // kept in sync with this order. Perhaps a better way is needed...
  648.             appendButton(box, "onPrefListUp",     { label: MSG_PREFS_MOVE_UP, 
  649.                                                     "class": "up" });
  650.             appendButton(box, "onPrefListDown",   { label: MSG_PREFS_MOVE_DOWN, 
  651.                                                     "class": "down" });
  652.             appendSeparator(box);
  653.             appendButton(box, "onPrefListAdd",    { label: MSG_PREFS_ADD });
  654.             appendButton(box, "onPrefListEdit",   { label: MSG_PREFS_EDIT });
  655.             appendButton(box, "onPrefListDelete", { label: MSG_PREFS_DELETE });
  656.             break;
  657.             
  658.         default:
  659.             // This is really more of an error case, since we really should
  660.             // know about all the valid pref types.
  661.             var label = document.createElement("label");
  662.             label.setAttribute("value", "[not editable] " + this.type);
  663.             this.box.appendChild(label);
  664.     }
  665.     
  666.     this.loadData();
  667.     
  668.     if (this.edit)
  669.     {
  670.         this.edit.setAttribute("prefobjectindex", this.parent.arrayIndex);
  671.         this.edit.setAttribute("prefname", this.name);
  672.     }
  673.     
  674.     if (!ASSERT("groups" in this.parent, "Must have called " +
  675.                 "[ObjectPrivateData].loadXUL before trying to display prefs."))
  676.         return;
  677.     
  678.     this.parent.addGroup(this.mainGroup);
  679.     if (this.subGroup)
  680.         this.parent.groups[this.mainGroup].addGroup(this.subGroup);
  681.     
  682.     if (!this.subGroup)
  683.         this.parent.groups[this.mainGroup].box.appendChild(this.box);
  684.     else
  685.         this.parent.groups[this.mainGroup].groups[this.subGroup].box.appendChild(this.box);
  686.     
  687.     // Setup tooltip stuff...
  688.     if (this.help && (this.type != "array"))
  689.     {
  690.         this.box.setAttribute("tooltiptitle", this.label);
  691.         this.box.setAttribute("tooltipcontent", this.help);
  692.         this.box.setAttribute("onmouseover", "gPrefWindow.onPrefMouseOver(this);");
  693.         this.box.setAttribute("onmousemove", "gPrefWindow.onPrefMouseMove(this);");
  694.         this.box.setAttribute("onmouseout", "gPrefWindow.onPrefMouseOut(this);");
  695.     }
  696. }
  697.  
  698. /* Loads the pref's data into the edit component. */
  699. PrefData.prototype.loadData =
  700. function pdata_loadData()
  701. {
  702.     /* Note about .value and .setAttribute as used here:
  703.      * 
  704.      * XBL doesn't kick in until CSS is calculated on a node, so the code makes 
  705.      * a compromise and uses these two methods as appropriate. Initally this 
  706.      * is called is before the node has been placed in the document DOM tree, 
  707.      * and thus hasn't been "magiced" by XBL and so .value is meaningless to 
  708.      * it. After initally being set as an attribute, it's added to the DOM, 
  709.      * XBL kicks in, and after that .value is the only way to change the value.
  710.      */
  711.     switch (this.type)
  712.     {
  713.         case "string":
  714.             if (this.edit.hasAttribute("value"))
  715.                 this.edit.value = this.val;
  716.             else
  717.                 this.edit.setAttribute("value", this.val);
  718.             break;
  719.             
  720.         case "number":
  721.             if (this.edit.hasAttribute("value"))
  722.                 this.edit.value = this.val;
  723.             else
  724.                 this.edit.setAttribute("value", this.val);
  725.             break;
  726.             
  727.         case "boolean":
  728.             if (this.edit.hasAttribute("checked"))
  729.                 this.edit.checked = this.val;
  730.             else
  731.                 this.edit.setAttribute("checked", this.val);
  732.             break;
  733.             
  734.         case "array":
  735.             // Remove old entires.
  736.             while (this.edit.firstChild)
  737.                 this.edit.removeChild(this.edit.firstChild);
  738.             
  739.             // Add new ones.
  740.             for (i = 0; i < this.val.length; i++)
  741.             {
  742.                 var item = document.createElement("listitem");
  743.                 item.value = this.val[i];
  744.                 item.crop = "center";
  745.                 item.setAttribute("label", this.val[i]);
  746.                 this.edit.appendChild(item);
  747.             }
  748.             
  749.             // Make sure buttons are up-to-date.
  750.             gPrefWindow.onPrefListSelect(this.edit);
  751.             break;
  752.             
  753.         default:
  754.     }
  755. }
  756.  
  757. /* Cleans up the mess. */
  758. PrefData.prototype.clear =
  759. function pdata_clear()
  760. {
  761.     //dd("Clearing pref " + this.name);
  762.     if (("box" in this) && this.box)
  763.     {
  764.         this.box.parentNode.removeChild(this.box);
  765.         delete this.box;
  766.     }
  767.     try {
  768.         this.manager.clearPref(this.name);
  769.     } catch(ex) {}
  770. }
  771.  
  772. /* Resets the pref to it's default. */
  773. PrefData.prototype.reset =
  774. function pdata_reset()
  775. {
  776.     //try {
  777.     //    this.manager.clearPref(this.name);
  778.     //} catch(ex) {}
  779.     this.val = this.def;
  780.     this.loadData();
  781. }
  782.  
  783. /* Saves the pref... or would do. */
  784. PrefData.prototype.save =
  785. function pdata_save()
  786. {
  787.     //FIXME//
  788. }
  789.  
  790. // Represents a "main group", i.e. a tab for a single pref object.
  791. function PrefMainGroup(parent, name)
  792. {
  793.     // Init this group's object.
  794.     this.parent = parent;  // Private data for pref object.
  795.     this.name = name;
  796.     this.groups = new Object();
  797.     this.label = getMsg("pref.group." + this.name + ".label", null, this.name);
  798.     this.tab = document.createElement("tab");
  799.     this.tabPanel = document.createElement("tabpanel");
  800.     this.box = this.sb = document.createElement("scroller");
  801.     
  802.     this.tab.setAttribute("label", this.label);
  803.     this.tabPanel.setAttribute("orient", "vertical");
  804.     this.sb.setAttribute("orient", "vertical");
  805.     this.sb.setAttribute("flex", 1);
  806.     
  807.     this.parent.tabs.appendChild(this.tab);
  808.     this.parent.tabPanels.appendChild(this.tabPanel);
  809.     this.tabPanel.appendChild(this.sb);
  810.     
  811.     return this;
  812. }
  813. // Adds a sub group to this main group.
  814. PrefMainGroup.prototype.addGroup =
  815. function pmgroup_addGroup(name)
  816. {
  817.     // If the sub group doesn't exist, make it exist.
  818.     if (!(name in this.groups))
  819.         this.groups[name] = new PrefSubGroup(this, name);
  820.     
  821.     return this.groups[name];
  822. }
  823.  
  824. // Represents a "sub group", i.e. a groupbox on a tab, for a single main group.
  825. function PrefSubGroup(parent, name)
  826. {
  827.     this.parent = parent;  // Main group.
  828.     this.name = name;
  829.     this.fullGroup = this.parent.name + "." + this.name;
  830.     this.label = getMsg("pref.group." + this.fullGroup + ".label", null, this.name);
  831.     this.help  = getMsg("pref.group." + this.fullGroup + ".help", null, "");
  832.     this.gb = document.createElement("groupbox");
  833.     this.cap = document.createElement("caption");
  834.     this.box = document.createElement("box");
  835.     
  836.     this.cap.setAttribute("label", this.label);
  837.     this.gb.appendChild(this.cap);
  838.     this.box.orient = "vertical";
  839.     
  840.     // If there's some help text, we place it as the first thing inside 
  841.     // the <groupbox>, as an explanation for the entire subGroup.
  842.     if (this.help)
  843.     {
  844.         this.helpLabel = document.createElement("label");
  845.         this.helpLabel.appendChild(document.createTextNode(this.help));
  846.         this.gb.appendChild(this.helpLabel);
  847.     }
  848.     this.gb.appendChild(this.box);
  849.     this.parent.box.appendChild(this.gb);
  850.     
  851.     return this;
  852. }
  853.  
  854. // Actual pref window itself.
  855. function PrefWindow()
  856. {
  857.     // Not loaded until the pref list and objects have been created in |onLoad|.
  858.     this.loaded = false;
  859.     
  860.     /* PREF TAB ORDER: Determins the order, and fixed tabs, found on the UI.
  861.      * 
  862.      * It is an array of mainGroup names, and a flag indicating if the tab
  863.      * should be created even when there's no prefs for it.
  864.      * 
  865.      * This is for consistency, although none of the ChatZilla built-in pref
  866.      * objects leave fixed tabs empty currently.
  867.      */
  868.     this.prefTabOrder = [{ fixed: true,  name: "general"}, 
  869.                          { fixed: true,  name: "appearance"}, 
  870.                          { fixed: false, name: "lists"}, 
  871.                          { fixed: false, name: "dcc"}, 
  872.                          { fixed: false, name: "startup"}, 
  873.                          { fixed: false, name: "global"}, 
  874.                          { fixed: false, name: "munger"}
  875.                         ];
  876.     
  877.     /* PREF OBJECTS: Stores all the objects we've using that have prefs.
  878.      * 
  879.      * Each object gets a "privateData" object, which is then used by the pref
  880.      * window code for storing all of it's data on the object.
  881.      */
  882.     this.prefObjects = null;
  883.     
  884.     /* TOOLTIPS: Special tooltips for preference items.
  885.      * 
  886.      * Timer:     return value from |setTimeout| whenever used. There is only 
  887.      *            ever one timer going for the tooltips.
  888.      * Showing:   stores visibility of the tooltip.
  889.      * ShowDelay: ms delay which them mouse must be still to show tooltips.
  890.      * HideDelay: ms delay before the tooltips hide themselves.
  891.      */
  892.     this.tooltipTimer = 0;
  893.     this.tooltipShowing = false;
  894.     this.tooltipShowDelay = 1000;
  895.     this.tooltipHideDelay = 20000;
  896. }
  897. PrefWindow.prototype.TYPE = "PrefWindow";
  898.  
  899. /* Updates the tooltip state, either showing or hiding it. */
  900. PrefWindow.prototype.setTooltipState =
  901. function pwin_setTooltipState(visible)
  902. {
  903.     // Shortcut: if we're already in the right state, don't bother.
  904.     if (this.tooltipShowing == visible)
  905.         return;
  906.     
  907.     var tt = document.getElementById("czPrefTip");
  908.     
  909.     // If we're trying to show it, and we have a reference object
  910.     // (this.tooltipObject), we are ok to go.
  911.     if (visible && this.tooltipObject)
  912.     {
  913.         /* Get the boxObject for the reference object, as we're going to
  914.          * place to tooltip explicitly based on this.
  915.          */
  916.         var tipobjBox = this.tooltipObject.boxObject;
  917.         
  918.         // Adjust the width to that of the reference box.
  919.         tt.sizeTo(tipobjBox.width, tt.boxObject.height);
  920.         /* show tooltip using the reference object, and it's screen 
  921.          * position. NOTE: Most of these parameters are supposed to affect
  922.          * things, and they do seem to matter, but don't work anything like
  923.          * the documentation. Be careful changing them.
  924.          */
  925.         tt.showPopup(this.tooltipObject, -1, -1, "tooltip", "bottomleft", "topleft");
  926.         
  927.         // Set the timer to hide the tooltip some time later...
  928.         // (note the fun inline function)
  929.         this.tooltipTimer = setTimeout(setTooltipState, this.tooltipHideDelay, 
  930.                                        this, false);
  931.         this.tooltipShowing = true;
  932.     }
  933.     else
  934.     {
  935.         /* We're here because either we are meant to be hiding the tooltip,
  936.          * or we lacked the information needed to show one. So hide it.
  937.          */
  938.         tt.hidePopup();
  939.         this.tooltipShowing = false;
  940.     }
  941. }
  942.  
  943. /** Window event handlers **/
  944.  
  945. /* Initalises, and loads all the data/utilities and prefs. */
  946. PrefWindow.prototype.onLoad =
  947. function pwin_onLoad()
  948. {
  949.     // Get ourselves a base object for the object hierarchy.
  950.     client = new PrefGlobal();
  951.     
  952.     // Kick off the localisation load.
  953.     initMessages();
  954.     
  955.     // Use localised name.
  956.     client.viewName = MSG_PREFS_GLOBAL;
  957.     client.unicodeName = client.viewName;
  958.     client.prettyName = client.viewName;
  959.     
  960.     // Use the window mediator service to prevent mutliple instances.
  961.     var windowMediator = Components.classes[MEDIATOR_CONTRACTID];
  962.     var windowManager = windowMediator.getService(nsIWindowMediator);
  963.     var enumerator = windowManager.getEnumerator(CONFIG_WINDOWTYPE);
  964.     
  965.     // We only want one open at a time because don't (currently) cope with
  966.     // pref-change notifications. In fact, it's not easy to cope with.
  967.     // Save it for some time later. :)
  968.     
  969.     enumerator.getNext();
  970.     if (enumerator.hasMoreElements())
  971.     {
  972.         alert(MSG_PREFS_ALREADYOPEN);
  973.         window.close();
  974.         return;
  975.     }
  976.     
  977.     // Kick off the core pref initalisation code.
  978.     initPrefs();
  979.     
  980.     // Turn off all notifications, or it'll get confused when any pref 
  981.     // does change.
  982.     client.prefManager.onPrefChanged = function(){};
  983.     
  984.     // The list of objects we're tacking the prefs of.
  985.     this.prefObjects = new PrefObjectList();
  986.     
  987.     /* Oh, this is such an odd way to do this. But hey, it works. :)
  988.      * What we do is ask the pref branch for the client object to give us
  989.      * a list of all the preferences under it. This just gets us all the
  990.      * Chatzilla prefs. Then, we filter them so that we only deal with
  991.      * ones for networks, channels and users. This means, even if only
  992.      * one pref is set for a channel, we get it's network and channel
  993.      * object created here.
  994.      */
  995.     var prefRE = new RegExp("^networks.([^.]+)" +
  996.                             "(?:\\.(users|channels)?\\.([^.]+))?\\.");
  997.     var rv = new Object();
  998.     var netList = client.prefManager.prefBranch.getChildList("networks.", rv);
  999.     for (i in netList)
  1000.     {
  1001.         var m = netList[i].match(prefRE);
  1002.         if (!m)
  1003.             continue;
  1004.         
  1005.         var netName = unMungeNetworkName(m[1]);
  1006.         /* We're forcing the network into existance (3rd param) if the
  1007.          * pref is actually set, as opposed to just being known about (the
  1008.          * pref branch will list all *known* prefs, not just those set).
  1009.          *
  1010.          * NOTE: |force| will, if |true|, set a property on the object so it
  1011.          *       will still exist next time we're here. If |false|, the
  1012.          *       the object will only come into existance if this magical
  1013.          *       [hidden] pref is set.
  1014.          */
  1015.         var force = client.prefManager.prefBranch.prefHasUserValue(netList[i]);
  1016.         
  1017.         // Don't bother with the new if it's already there (time saving).
  1018.         if (!(netName in client.networks))
  1019.             new PrefNetwork(client, netName, force);
  1020.         
  1021.         if ((2 in m) && (3 in m) && (netName in client.networks))
  1022.         {
  1023.             var net = client.networks[netName];
  1024.             
  1025.             // Create a channel object if appropriate.
  1026.             if (m[2] == "channels")
  1027.                 new PrefChannel(net.primServ, unMungeNetworkName(m[3]), force);
  1028.             
  1029.             // Create a user object if appropriate.
  1030.             if (m[2] == "users")
  1031.                 new PrefUser(net.primServ, unMungeNetworkName(m[3]), force);
  1032.         }
  1033.     }
  1034.     
  1035.     // Store out object that represents the current view.
  1036.     var currentView = null;
  1037.     
  1038.     // Enumerate source window's tab list...
  1039.     if (("arguments" in window) && (0 in window.arguments) && ("client" in window.arguments[0]))
  1040.     {
  1041.         /* Make sure we survive this, external data could be bad. :) */
  1042.         try {
  1043.             var czWin = window.arguments[0];
  1044.             var s;
  1045.             var n, c, u;
  1046.             
  1047.             /* Go nick the source window's view list. We can then show items in
  1048.              * the tree for the currently connected/shown networks, channels
  1049.              * and users even if they don't have any known prefs yet.
  1050.              * 
  1051.              * NOTE: the |false, true| params tell the objects to not force 
  1052.              *       any prefs into existance, but still show in the tree.
  1053.              */
  1054.             for (i = 0; i < czWin.client.viewsArray.length; i++)
  1055.             {
  1056.                 var view = czWin.client.viewsArray[i].source;
  1057.                 
  1058.                 // Network view...
  1059.                 if (view.TYPE == "IRCNetwork")
  1060.                 {
  1061.                     n = new PrefNetwork(client, view.unicodeName, false, true);
  1062.                     if (view == czWin.client.currentObject)
  1063.                         currentView = n;
  1064.                 }
  1065.                 
  1066.                 if (view.TYPE.match(/^IRC(Channel|User)$/))
  1067.                     s = client.networks[view.parent.parent.canonicalName].primServ;
  1068.                 
  1069.                 // Channel view...
  1070.                 if (view.TYPE == "IRCChannel")
  1071.                 {
  1072.                     c = new PrefChannel(s, view.unicodeName, false, true);
  1073.                     if (view == czWin.client.currentObject)
  1074.                         currentView = c;
  1075.                 }
  1076.                 
  1077.                 // User view...
  1078.                 if (view.TYPE == "IRCUser")
  1079.                 {
  1080.                     u = new PrefUser(s, view.nick, false, true);
  1081.                     if (view == czWin.client.currentObject)
  1082.                         currentView = u;
  1083.                 }
  1084.             }
  1085.         } catch(ex) {}
  1086.     }
  1087.     
  1088.     // Add the client object...
  1089.     this.prefObjects.addObject(client);
  1090.     // ...and everyone else.
  1091.     var i, j;
  1092.     /* We sort the keys (property names, i.e. network names). This means the UI
  1093.      * will show them in lexographical order, which is good.
  1094.      */
  1095.     var sortedNets = keys(client.networks).sort();
  1096.     for (i = 0; i < sortedNets.length; i++) {
  1097.         net = client.networks[sortedNets[i]];
  1098.         this.prefObjects.addObject(net);
  1099.         
  1100.         var sortedChans = keys(net.channels).sort();
  1101.         for (j = 0; j < sortedChans.length; j++)
  1102.             this.prefObjects.addObject(net.channels[sortedChans[j]]);
  1103.         
  1104.         var sortedUsers = keys(net.users).sort();
  1105.         for (j = 0; j < sortedUsers.length; j++)
  1106.             this.prefObjects.addObject(net.users[sortedUsers[j]]);
  1107.     }
  1108.     
  1109.     // Select the first item in the list.
  1110.     var prefTree = document.getElementById("pref-tree-object");
  1111.     if ("selection" in prefTree.treeBoxObject)
  1112.         prefTree.treeBoxObject.selection.select(0);
  1113.     else
  1114.         prefTree.view.selection.select(0);
  1115.     
  1116.     // Find and select the current view.
  1117.     for (i = 0; i < this.prefObjects.objects.length; i++)
  1118.     {
  1119.         if (this.prefObjects.objects[i] == currentView)
  1120.         {
  1121.             if ("selection" in prefTree.treeBoxObject)
  1122.                 prefTree.treeBoxObject.selection.select(i);
  1123.             else
  1124.                 prefTree.view.selection.select(i);
  1125.         }
  1126.     }
  1127.     
  1128.     this.onSelectObject();
  1129.     
  1130.     // We're done, without error, so it's safe to show the stuff.
  1131.     document.getElementById("loadDeck").selectedIndex = 1;
  1132.     // This allows [OK] to actually save, without this it'll just close.
  1133.     this.loaded = true;
  1134.     
  1135.     // Force the window to be the right size now, not later.
  1136.     window.sizeToContent();
  1137.     
  1138.     // Center window.
  1139.     if (("arguments" in window) && (0 in window.arguments))
  1140.     {
  1141.         var ow = window.arguments[0];
  1142.         
  1143.         window.moveTo(ow.screenX + Math.max((ow.outerWidth  - window.outerWidth ) / 2, 0), 
  1144.                       ow.screenY + Math.max((ow.outerHeight - window.outerHeight) / 2, 0));
  1145.     }
  1146. }
  1147.  
  1148. /* Closing the window. Clean up. */
  1149. PrefWindow.prototype.onClose =
  1150. function pwin_onClose()
  1151. {
  1152.     if (this.loaded)
  1153.         destroyPrefs();
  1154. }
  1155.  
  1156. /* OK button. */
  1157. PrefWindow.prototype.onOK =
  1158. function pwin_onOK()
  1159. {
  1160.     if (this.onApply())
  1161.         window.close();
  1162.     return true;
  1163. }
  1164.  
  1165. /* Apply button. */
  1166. PrefWindow.prototype.onApply =
  1167. function pwin_onApply()
  1168. {
  1169.     // If the load failed, better not to try to save.
  1170.     if (!this.loaded)
  1171.         return false;
  1172.     
  1173.     try {
  1174.         // Get an array of all the (XUL) items we have to save.
  1175.         var list = getPrefTags();
  1176.         
  1177.         //if (!confirm("There are " + list.length + " pref tags to save. OK?")) return false;
  1178.         
  1179.         for (var i = 0; i < list.length; i++)
  1180.         {
  1181.             // Save this one pref...
  1182.             var index  = list[i].getAttribute("prefobjectindex");
  1183.             var name   = list[i].getAttribute("prefname");
  1184.             // Get the private data for the object out, since everything is
  1185.             // based on this.
  1186.             var po     = this.prefObjects.getObject(index);
  1187.             var pref   = po.prefs[name];
  1188.             
  1189.             var value;
  1190.             // We just need to force the value from the DOMString form to
  1191.             // the right form, so we don't get anything silly happening.
  1192.             switch (pref.type)
  1193.             {
  1194.                 case "string":
  1195.                     value = list[i].value;
  1196.                     break;
  1197.                 case "number":
  1198.                     value = 1 * list[i].value;
  1199.                     break;
  1200.                 case "boolean":
  1201.                     value = list[i].checked;
  1202.                     break;
  1203.                 case "array":
  1204.                     var l = new Array();
  1205.                     for (var j = 0; j < list[i].childNodes.length; j++)
  1206.                         l.push(list[i].childNodes[j].value);
  1207.                     value = l;
  1208.                     break;
  1209.                 default:
  1210.                     throw "Unknown pref type: " + pref.type + "!";
  1211.             }
  1212.             /* Fun stuff. We don't want to save if the user hasn't changed
  1213.              * it. This is so that if the default is defered, and changed,
  1214.              * but this hasn't, we keep the defered default. Which is a
  1215.              * Good Thing. :)
  1216.              */
  1217.             if (((pref.type != "array") && (value != pref.startVal)) || 
  1218.                 ((pref.type == "array") && 
  1219.                  (value.join(";") != pref.startVal.join(";"))))
  1220.             {
  1221.                 po.parent.prefs[pref.name] = value;
  1222.             }
  1223.             // Update our saved value, so the above check works the 2nd 
  1224.             // time the user clicks Apply.
  1225.             pref.val = value;
  1226.             pref.startVal = pref.val;
  1227.         }
  1228.         
  1229.         return true;
  1230.     }
  1231.     catch (e)
  1232.     {
  1233.         alert(getMsg(MSG_PREFS_ERR_SAVE, e));
  1234.         return false;
  1235.     }
  1236. }
  1237.  
  1238. /* Cancel button. */
  1239. PrefWindow.prototype.onCancel =
  1240. function pwin_onCancel()
  1241. {
  1242.     window.close();
  1243.     return true;
  1244. }
  1245.  
  1246. /** Tooltips' event handlers **/
  1247.  
  1248. PrefWindow.prototype.onPrefMouseOver =
  1249. function pwin_onPrefMouseOver(object)
  1250. {
  1251.     this.tooltipObject = object;
  1252.     this.tooltipTitle = object.getAttribute("tooltiptitle");
  1253.     this.tooltipText = object.getAttribute("tooltipcontent");
  1254.     // Reset the timer now we're over a new pref.
  1255.     clearTimeout(this.tooltipTimer);
  1256.     this.tooltipTimer = setTimeout(setTooltipState, this.tooltipShowDelay, 
  1257.                                    this, true);
  1258. }
  1259.  
  1260. PrefWindow.prototype.onPrefMouseMove =
  1261. function pwin_onPrefMouseMove(object)
  1262. {
  1263.     // If the tooltip isn't showing, we need to reset the timer.
  1264.     if (!this.tooltipShowing)
  1265.     {
  1266.         clearTimeout(this.tooltipTimer);
  1267.         this.tooltipTimer = setTimeout(setTooltipState, this.tooltipShowDelay, 
  1268.                                        this, true);
  1269.     }
  1270. }
  1271.  
  1272. PrefWindow.prototype.onPrefMouseOut =
  1273. function pwin_onPrefMouseOut(object)
  1274. {
  1275.     // Left the pref! Hide tooltip, and clear timer.
  1276.     this.setTooltipState(false);
  1277.     clearTimeout(this.tooltipTimer);
  1278. }
  1279.  
  1280. PrefWindow.prototype.onTooltipPopupShowing =
  1281. function pwin_onTooltipPopupShowing(popup)
  1282. {
  1283.     if (!this.tooltipText)
  1284.         return false;
  1285.     
  1286.     var fChild = popup.firstChild;
  1287.     var diff = popup.boxObject.height - fChild.boxObject.height;
  1288.     
  1289.     // Setup the labels...
  1290.     var ttt = document.getElementById("czPrefTipTitle");
  1291.     ttt.firstChild.nodeValue = this.tooltipTitle;
  1292.     var ttl = document.getElementById("czPrefTipLabel");
  1293.     ttl.firstChild.nodeValue = this.tooltipText;
  1294.     
  1295.     popup.sizeTo(popup.boxObject.width, fChild.boxObject.height + diff);
  1296.     
  1297.     return true;
  1298. }
  1299.  
  1300. /** Components' event handlers **/
  1301.  
  1302. // Selected an item in the tree.
  1303. PrefWindow.prototype.onSelectObject =
  1304. function pwin_onSelectObject()
  1305. {
  1306.     var prefTree = document.getElementById("pref-tree-object");
  1307.     var rv = new Object();
  1308.     if ("selection" in prefTree.treeBoxObject)
  1309.         prefTree.treeBoxObject.selection.getRangeAt(0, rv, {});
  1310.     else
  1311.         prefTree.view.selection.getRangeAt(0, rv, {});
  1312.     var prefTreeIndex = rv.value;
  1313.     
  1314.     var delButton = document.getElementById("object-delete");
  1315.     if (prefTreeIndex > 0)
  1316.         delButton.removeAttribute("disabled");
  1317.     else
  1318.         delButton.setAttribute("disabled", "true");
  1319.     
  1320.     var prefItem = prefTree.contentView.getItemAtIndex(prefTreeIndex);
  1321.     
  1322.     var pObjectIndex = prefItem.getAttribute("prefobjectindex");
  1323.     var pObject = this.prefObjects.getObject(pObjectIndex);
  1324.     
  1325.     if (!ASSERT(pObject, "Object not found for index! " + 
  1326.                 prefItem.getAttribute("prefobjectindex")))
  1327.         return;
  1328.     
  1329.     pObject.loadXUL(this.prefTabOrder);
  1330.     
  1331.     var header = document.getElementById("pref-header");
  1332.     header.setAttribute("title", getMsg(MSG_PREFS_FMT_HEADER, 
  1333.                                         pObject.parent.prettyName));
  1334.     
  1335.     var deck = document.getElementById("pref-object-deck");
  1336.     deck.selectedIndex = pObject.deckIndex;
  1337.     
  1338.     this.currentObject = pObject;
  1339. }
  1340.  
  1341. // Browse button for file prefs.
  1342. PrefWindow.prototype.onPrefBrowse =
  1343. function pwin_onPrefBrowse(button)
  1344. {
  1345.     var spec = "$all";
  1346.     if (button.hasAttribute("spec"))
  1347.         spec = button.getAttribute("spec") + " " + spec;
  1348.     
  1349.     var rv = pickOpen(MSG_PREFS_BROWSE_TITLE, spec);
  1350.     if (rv.reason != PICK_OK)
  1351.         return;
  1352.     
  1353.     var data = { file: rv.file.path, fileurl: rv.picker.fileURL.spec };
  1354.     var edit = button.previousSibling.lastChild;
  1355.     var type = button.getAttribute("kind");
  1356.     edit.value = data[type];
  1357. },
  1358.  
  1359. // Selection changed on listbox.
  1360. PrefWindow.prototype.onPrefListSelect =
  1361. function pwin_onPrefListSelect(object)
  1362. {
  1363.     var list = getRelatedItem(object, "list");
  1364.     var buttons = new Object();
  1365.     buttons.up   = getRelatedItem(object, "button-up");
  1366.     buttons.down = getRelatedItem(object, "button-down");
  1367.     buttons.add  = getRelatedItem(object, "button-add");
  1368.     buttons.edit = getRelatedItem(object, "button-edit");
  1369.     buttons.del  = getRelatedItem(object, "button-delete");
  1370.     
  1371.     if (("selectedItems" in list) && list.selectedItems && 
  1372.         list.selectedItems.length)
  1373.     {
  1374.         buttons.edit.removeAttribute("disabled");
  1375.         buttons.del.removeAttribute("disabled");
  1376.     }
  1377.     else
  1378.     {
  1379.         buttons.edit.setAttribute("disabled", "true");
  1380.         buttons.del.setAttribute("disabled", "true");
  1381.     }
  1382.     
  1383.     if (!("selectedItems" in list) || !list.selectedItems || 
  1384.         list.selectedItems.length == 0 || list.selectedIndex == 0)
  1385.     {
  1386.         buttons.up.setAttribute("disabled", "true");
  1387.     }
  1388.     else
  1389.     {
  1390.         buttons.up.removeAttribute("disabled");
  1391.     }
  1392.     
  1393.     if (!("selectedItems" in list) || !list.selectedItems || 
  1394.         list.selectedItems.length == 0 || 
  1395.         list.selectedIndex == list.childNodes.length - 1)
  1396.     {
  1397.         buttons.down.setAttribute("disabled", "true");
  1398.     }
  1399.     else
  1400.     {
  1401.         buttons.down.removeAttribute("disabled");
  1402.     }
  1403. }
  1404.  
  1405. // Up button for lists.
  1406. PrefWindow.prototype.onPrefListUp =
  1407. function pwin_onPrefListUp(object)
  1408. {
  1409.     var list = getRelatedItem(object, "list");
  1410.     
  1411.     var selected = list.selectedItems[0];
  1412.     var before = selected.previousSibling;
  1413.     if (before)
  1414.     {
  1415.         before.parentNode.insertBefore(selected, before);
  1416.         list.selectItem(selected);
  1417.         list.ensureElementIsVisible(selected);
  1418.     }
  1419. }
  1420.  
  1421. // Down button for lists.
  1422. PrefWindow.prototype.onPrefListDown =
  1423. function pwin_onPrefListDown(object)
  1424. {
  1425.     var list = getRelatedItem(object, "list");
  1426.     
  1427.     var selected = list.selectedItems[0];
  1428.     if (selected.nextSibling)
  1429.     {
  1430.         if (selected.nextSibling.nextSibling)
  1431.             list.insertBefore(selected, selected.nextSibling.nextSibling);
  1432.         else
  1433.             list.appendChild(selected);
  1434.         
  1435.         list.selectItem(selected);
  1436.     }
  1437. }
  1438.  
  1439. // Add button for lists.
  1440. PrefWindow.prototype.onPrefListAdd =
  1441. function pwin_onPrefListAdd(object)
  1442. {
  1443.     var list = getRelatedItem(object, "list");
  1444.     var newItem;
  1445.     
  1446.     switch (list.getAttribute("kind"))
  1447.     {
  1448.         case "url":
  1449.             var item = prompt(MSG_PREFS_LIST_ADD);
  1450.             if (item)
  1451.             {
  1452.                 newItem = document.createElement("listitem");
  1453.                 newItem.setAttribute("label", item);
  1454.                 newItem.value = item;
  1455.                 list.appendChild(newItem);
  1456.                 this.onPrefListSelect(object);
  1457.             }
  1458.             break;
  1459.         case "file":
  1460.         case "fileurl":
  1461.             var spec = "$all";
  1462.             
  1463.             var rv = pickOpen(MSG_PREFS_BROWSE_TITLE, spec);
  1464.             if (rv.reason == PICK_OK)
  1465.             {
  1466.                 var data = { file: rv.file.path, fileurl: rv.picker.fileURL.spec };
  1467.                 var kind = list.getAttribute("kind");
  1468.                 
  1469.                 newItem = document.createElement("listitem");
  1470.                 newItem.setAttribute("label", data[kind]);
  1471.                 newItem.value = data[kind];
  1472.                 list.appendChild(newItem);
  1473.                 this.onPrefListSelect(object);
  1474.             }
  1475.             
  1476.             break;
  1477.     }
  1478. }
  1479.  
  1480. // Edit button for lists.
  1481. PrefWindow.prototype.onPrefListEdit =
  1482. function pwin_onPrefListEdit(object)
  1483. {
  1484.     var list = getRelatedItem(object, "list");
  1485.     
  1486.     switch (list.getAttribute("kind"))
  1487.     {
  1488.         case "url":
  1489.         case "file":
  1490.         case "fileurl":
  1491.             // We're letting the user edit file types here, since it saves us 
  1492.             // a lot of work, and we can't let them pick a file OR a directory,
  1493.             // so they pick a file and can edit it off to use a directory.
  1494.             var listItem = list.selectedItems[0];
  1495.             var newValue = prompt(MSG_PREFS_LIST_EDIT, listItem.value);
  1496.             if (newValue)
  1497.             {
  1498.                 listItem.setAttribute("label", newValue);
  1499.                 listItem.value = newValue;
  1500.             }
  1501.             break;
  1502.     }
  1503. }
  1504.  
  1505. // Delete button for lists.
  1506. PrefWindow.prototype.onPrefListDelete =
  1507. function pwin_onPrefListDelete(object)
  1508. {
  1509.     var list = getRelatedItem(object, "list");
  1510.     
  1511.     var listItem = list.selectedItems[0];
  1512.     if (confirm(getMsg(MSG_PREFS_LIST_DELETE, listItem.value)))
  1513.         list.removeChild(listItem);
  1514. }
  1515.  
  1516. /* Add... button. */
  1517. PrefWindow.prototype.onAddObject =
  1518. function pwin_onAddObject()
  1519. {
  1520.     var rv = new Object();
  1521.     
  1522.     /* Try to nobble the current selection and pre-fill as needed. */
  1523.     switch (this.currentObject.parent.TYPE)
  1524.     {
  1525.         case "PrefNetwork":
  1526.             rv.type = "net";
  1527.             rv.net = this.currentObject.parent.unicodeName;
  1528.             break;
  1529.         case "PrefChannel":
  1530.             rv.type = "chan";
  1531.             rv.net = this.currentObject.parent.parent.parent.unicodeName;
  1532.             rv.chan = this.currentObject.parent.unicodeName;
  1533.             break;
  1534.         case "PrefUser":
  1535.             rv.type = "user";
  1536.             rv.net = this.currentObject.parent.parent.parent.unicodeName;
  1537.             rv.chan = this.currentObject.parent.unicodeName;
  1538.             break;
  1539.     }
  1540.     
  1541.     // Show add dialog, passing the data object along.
  1542.     window.openDialog("config-add.xul", "cz-config-add", "chrome,dialog,modal", rv);
  1543.     
  1544.     if (!rv.ok)
  1545.         return;
  1546.     
  1547.     /* Ok, so what type did they want again?
  1548.      * 
  1549.      * NOTE: The param |true| in the object creation calls is for |force|. It
  1550.      *       causes the hidden pref to be set for the objects so they are shown
  1551.      *       every time this window opens, until the user deletes them.
  1552.      */
  1553.     switch (rv.type)
  1554.     {
  1555.         case "net":
  1556.             this.prefObjects.addObject(new PrefNetwork(client, rv.net, true));
  1557.             break;
  1558.         case "chan":
  1559.             if (!(rv.net in client.networks))
  1560.                 this.prefObjects.addObject(new PrefNetwork(client, rv.net, true));
  1561.             this.prefObjects.addObject(new PrefChannel(client.networks[rv.net].primServ, rv.chan, true));
  1562.             break;
  1563.         case "user":
  1564.             if (!(rv.net in client.networks))
  1565.                 this.prefObjects.addObject(new PrefNetwork(client, rv.net, true));
  1566.             this.prefObjects.addObject(new PrefUser(client.networks[rv.net].primServ, rv.chan, true));
  1567.             break;
  1568.         default:
  1569.             // Oops. Not good, if we got here.
  1570.             alert("Unknown pref type: " + rv.type);
  1571.     }
  1572. }
  1573.  
  1574. /* Delete button. */
  1575. PrefWindow.prototype.onDeleteObject =
  1576. function pwin_onDeleteObject()
  1577. {
  1578.     // Save current node before we re-select.
  1579.     var sel = this.currentObject;
  1580.     
  1581.     // Check they want to go ahead.
  1582.     if (!confirm(getMsg(MSG_PREFS_OBJECT_DELETE, sel.parent.unicodeName)))
  1583.         return;
  1584.     
  1585.     // Select a new item BEFORE removing the current item, so the <tree> 
  1586.     // doesn't freak out on us.
  1587.     var prefTree = document.getElementById("pref-tree-object");
  1588.     if ("selection" in prefTree.treeBoxObject)
  1589.         prefTree.treeBoxObject.selection.select(0);
  1590.     else
  1591.         prefTree.view.selection.select(0);
  1592.     
  1593.     // If it's a network, nuke all the channels and users too.
  1594.     if (sel.parent.TYPE == "PrefNetwork")
  1595.     {
  1596.         var chans = sel.parent.channels;
  1597.         for (k in chans)
  1598.             PrefObjectList.getPrivateData(chans[k]).clear();
  1599.         
  1600.         var users = sel.parent.users;
  1601.         for (k in users)
  1602.             PrefObjectList.getPrivateData(users[k]).clear();
  1603.     }
  1604.     sel.clear();
  1605.     
  1606.     this.onSelectObject();
  1607. }
  1608.  
  1609. /* Reset button. */
  1610. PrefWindow.prototype.onResetObject =
  1611. function pwin_onResetObject()
  1612. {
  1613.     // Save current node before we re-select.
  1614.     var sel = this.currentObject;
  1615.     
  1616.     // Check they want to go ahead.
  1617.     if (!confirm(getMsg(MSG_PREFS_OBJECT_RESET, sel.parent.unicodeName)))
  1618.         return;
  1619.     
  1620.     // Reset the prefs.
  1621.     sel.reset();
  1622. }
  1623.  
  1624. // End of PrefWindow. //
  1625.  
  1626. /*** Base functions... ***/
  1627.  
  1628. /* Gets a "related" items, such as the buttons associated with a list. */
  1629. function getRelatedItem(object, thing)
  1630. {
  1631.     switch (object.nodeName)
  1632.     {
  1633.         case "listbox":
  1634.             switch (thing) {
  1635.                 case "list":
  1636.                     return object;
  1637.                 case "button-up":
  1638.                     return object.parentNode.nextSibling.childNodes[0];
  1639.                 case "button-down":
  1640.                     return object.parentNode.nextSibling.childNodes[1];
  1641.                 case "button-add":
  1642.                     return object.parentNode.nextSibling.childNodes[3];
  1643.                 case "button-edit":
  1644.                     return object.parentNode.nextSibling.childNodes[4];
  1645.                 case "button-delete":
  1646.                     return object.parentNode.nextSibling.childNodes[5];
  1647.             }
  1648.             break;
  1649.         case "button":
  1650.             var n = object.parentNode.previousSibling.lastChild;
  1651.             if (n)
  1652.                 return getRelatedItem(n, thing);
  1653.             break;
  1654.     }
  1655.     return null;
  1656. }
  1657.  
  1658. // Wrap this call so we have the right |this|.
  1659. function setTooltipState(w, s)
  1660. {
  1661.     w.setTooltipState(s);
  1662. }
  1663.  
  1664. // Reverses the Pref Manager's munging of network names.
  1665. function unMungeNetworkName(name)
  1666. {
  1667.     name = ecmaUnescape(name);
  1668.     return name.replace(/_/g, ":").replace(/-/g, ".");
  1669. }
  1670.  
  1671. // Adds a button to a container, setting up the command in a simple way.
  1672. function appendButton(cont, oncommand, attr)
  1673. {
  1674.     var btn = document.createElement("button");
  1675.     if (attr)
  1676.         for (var a in attr)
  1677.             btn.setAttribute(a, attr[a]);
  1678.     if (oncommand)
  1679.         btn.setAttribute("oncommand", "gPrefWindow." + oncommand + "(this);");
  1680.     else
  1681.         btn.setAttribute("disabled", "true");
  1682.     cont.appendChild(btn);
  1683. }
  1684.  
  1685. // Like appendButton, but just drops in a separator.
  1686. function appendSeparator(cont, attr)
  1687. {
  1688.     var spacer = document.createElement("separator");
  1689.     if (attr)
  1690.         for (var a in attr)
  1691.             spacer.setAttribute(a, attr[a]);
  1692.     cont.appendChild(spacer);
  1693. }
  1694.  
  1695. /* This simply collects together all the <textbox>, <checkbox> and <listbox> 
  1696.  * elements that have the attribute "prefname". Thus, we generate a list of 
  1697.  * all elements that are for prefs.
  1698.  */
  1699. function getPrefTags()
  1700. {
  1701.     var rv = new Array();
  1702.     var i, list;
  1703.     
  1704.     list = document.getElementsByTagName("textbox");
  1705.     for (i = 0; i < list.length; i++)
  1706.     {
  1707.         if (list[i].hasAttribute("prefname"))
  1708.             rv.push(list[i]);
  1709.     }
  1710.     list = document.getElementsByTagName("checkbox");
  1711.     for (i = 0; i < list.length; i++)
  1712.     {
  1713.         if (list[i].hasAttribute("prefname"))
  1714.             rv.push(list[i]);
  1715.     }
  1716.     list = document.getElementsByTagName("listbox");
  1717.     for (i = 0; i < list.length; i++)
  1718.     {
  1719.         if (list[i].hasAttribute("prefname"))
  1720.             rv.push(list[i]);
  1721.     }
  1722.     
  1723.     return rv;
  1724. }
  1725.  
  1726. // Sets up the "extra1" button (Apply).
  1727. function setupButtons()
  1728. {
  1729.     // Hacky-hacky-hack. Looks like the new toolkit does provide a solution,
  1730.     // but we need to support SeaMonkey too. :)
  1731.     
  1732.     var dialog = document.documentElement;
  1733.     dialog.getButton("extra1").label = dialog.getAttribute("extra1Label");
  1734. }
  1735.  
  1736. // And finally, we want one of these.
  1737. var gPrefWindow = new PrefWindow();
  1738.